<!DOCTYPE HTML>
<html>
<head>
<title>PostMessage / SendMessage 教程 | AutoHotkey</title>
<meta name="description" content="Learn how to send messages to a window or its controls using the PostMessage or SendMessage commands." />
<meta http-equiv="content-type" content="text/html; charset=utf-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<link href="../static/theme.css" rel="stylesheet" type="text/css" />
<script src="../static/content.js" type="text/javascript"></script>
<script type="text/javascript">$(function(){0<=window.navigator.userAgent.toLowerCase().indexOf("ucbrowser")&&CaoNiMaDeUc()})</script>
</head>

<body>
<h1>PostMessage / SendMessage 教程 <span class="headnote">作者: Rajat</span></h1>

<p>本页解释如何通过 <a href="../commands/PostMessage.htm">PostMessage</a> 和 <a href="../commands/PostMessage.htm">SendMessage</a> 向窗口或窗口控件发送消息, 并将回答如下问题:</p>
<p>"如何按下已最小化窗口中的按钮?"<br>
"当 <a href="../commands/WinMenuSelectItem.htm">WinMenuSelectItem</a> 不起作用时, 我如何选择一个菜单项?!"<br>
"这是个可更换皮肤的窗口.... 如何确保我每次发送的命令都有效?"<br>
"如果是 <strong>隐藏</strong> 窗口呢?!"<br>
<br>
需要: <span class="ver">[AutoHotkey v1.0.09+]</span> 和 Winspector Spy(<a href="https://www.softpedia.com/get/Security/Security-Related/Winspector.shtml">点击打开下载页面</a>)</p>
<p>像第一个例子那样, 请注意 <a href="../commands/WinMenuSelectItem.htm">WinMenuSelectItem</a> 对 Outlook Express 的 "New Message" 窗口中的菜单栏无效. 换句话说, 这代码不起作用:</p>
<pre>WinMenuSelectItem, New Message,, &amp;Insert, &amp;Picture...</pre>
<p>但 <a href="../commands/PostMessage.htm">PostMessage</a> 能实现这个操作:</p>
<pre>PostMessage, 0x111, 40239, 0, , New Message</pre>
<p>太神奇了!但那个是什么? 0x111 是 <a href="SendMessageList.htm">wm_command 消息</a>的十六进制代码, 而 40239 是这个特殊窗口理解为选择 'Insert Picture' 菜单项的代码. 现在让我告诉您如何找到类似 40239 这样的值:</p>
<ol>
  <li>启动 Winspector Spy, 打开 "New Message" 窗口.</li>
  <li>拖动 Winspector Spy 窗口中的十字线到 "New Message" 窗口的标题栏(没有被 Winspector Spy 覆盖的部分).</li>
  <li>在左边的列表中右键点击所选窗口并选择 'Messages'.</li>
  <li>右键点击空白窗口并选择 'Edit message filter'.</li>
  <li>点击 'filter all' 按钮, 然后双击左边列表中的 'wm_command'. 这样您将只监视此消息.</li>
  <li>现在转到 "New Message" 窗口并在菜单栏中选择: Insert &gt; Picture.</li>
  <li>返回 Winspector Spy 并按下信号灯按钮来停止监视.</li>
  <li>展开收集到的 wm_command 消息(忽略其他的).</li>
  <li>您想要找的(通常) 是代码为 0 的消息. 有时会有一些描述为 'win activated' 或 'win destroyed' 以及其他..的 wm_command 消息都是不需要的内容. 您会发现描述为 'Control ID: 40239' 的消息 ...就是它了!</li>
  <li>现在把它放入上面的命令中您就完成了! 它就是 wParam 值.</li>
</ol>
<p>在下一个例子中我会使用画图程序, 因为很可能每个人都会有. 现在假设我们要使用 AutoHotkey 从工具栏中选择一个工具; 假设要选择取色工具.</p>
<p>您会怎么做? 很可能是使用鼠标点击工具栏, 对吗? 但是按钮可能被移动且隐藏了!这个画图程序的工具栏也可能被移动/隐藏. 所以如果目标用户这么做了, 那么您的脚本在这个点上会失效. 但是下面的命令仍然有效:</p>
<pre>PostMessage, 0x111, 639,,, Untitled - Paint</pre>
<p><a href="../commands/PostMessage.htm">PostMessage</a> 的另一个好处是窗口可以在后台; 与之相比, 发送鼠标点击要求目标窗口必须是活动的.</p>
<p>这里有其他的例子. 请注意: 我使用的是 WinXP Pro(SP1) ... 如果您使用不同的操作系统, 那么您的参数可能要改变(仅适用于像写字板和记事本这类 windows 自带的应用程序; 其他的程序应该不用改变):</p>
<pre><em>; 设置写字板字体为青色</em>
<a href="../commands/PostMessage.htm">PostMessage</a>, 0x111, 32788, 0, , Document - WordPad</pre>
<pre><em>; 打开记事本的关于对话框</em>
<a href="../commands/PostMessage.htm">PostMessage</a>, 0x111, 65, 0, , Untitled - Notepad</pre>
<pre><em>; 切换记事本的自动换行属性</em>
<a href="../commands/PostMessage.htm">PostMessage</a>, 0x111, 32, 0, , Untitled - Notepad</pre>
<pre><em>; 在 Windows Media Player 中播放/暂停</em>
<a href="../commands/PostMessage.htm">PostMessage</a>, 0x111, 32808, 0, , Windows Media Player</pre>
<pre><em>; 挂起正在运行的 AHK 脚本的热键</em>
DetectHiddenWindows, On
SetTitleMatchMode, 2
<em>; 使用 65306 来 Pause, 而 65303 来 Reload, 从而代替下面的 Suspend. (请参阅 <a href="../FAQ.htm#close">FAQ</a>)</em>
<a href="../commands/PostMessage.htm">PostMessage</a>, 0x111, 65305,,, MyScript.ahk - AutoHotkey
</pre>
<pre><em>; 按下 CapsLock 和 Numpad2 重新加载所有 AutoHotkey 脚本</em>
CapsLock & Numpad2::
ReloadAllAhkScripts() {
    DetectHiddenWindows, On
    SetTitleMatchMode, 2
    WinGet, allAhkExe, List, ahk_class AutoHotkey
    Loop, % allAhkExe {
        hwnd := allAhkExe%A_Index%
        if (hwnd = A_ScriptHwnd)  <em>; 重新加载时忽略当前窗口</em>
        {
            continue
        }
        PostMessage, 0x111, 65303,,, % "ahk_id" . hwnd
    }
    Reload
}
</pre>
<p>上面讲的是 PostMessage. <a href="../commands/PostMessage.htm">SendMessage</a> 的工作方式相同, 不过还会等待返回值, 这个值可用于一些情况中, 例如获取 Winamp 中当前正在播放的曲目(请参阅<a href="Winamp.htm">自动化 Winamp</a> 的例子).</p>
<p>这里是一些注意事项:</p>
<ul>
  <li>上文也提到了操作系统是 XP 且需要注意消息值随着不同的操作系统而改变. 如果您发现一个消息在您的系统上有效(对于一个软件的某个版本), 那么在其他系统上它对于这个软件的相同版本仍然是有效的. 此外, 大多数应用程序甚至会在它们的不同版本中使用相同的消息值(例如 Windows Media Player 和 Winamp).</li>
  <li>如果您已经在 Winspector Spy 中设置了过滤器来只显示 wm_command 消息, 却仍然得到大量的消息时, 请右键点击那些不需要的消息并选择隐藏(消息名)... 这些消息一般是不是由于您和目标软件交互所产生的消息.</li>
  <li>在 Winspector Spy 中向右的箭头显示消息值, 而模糊的左箭头显示返回值. 返回值 0(零) 默认情况下可以安全地认为 '没有错误'(与 SendMessage 一起使用, 返回值会存储在 <a href="ErrorLevel.htm">%ErrorLevel%</a>).</li>
  <li>要传递消息到匹配部分标题的窗口, 请添加此行到脚本: <br><code>SetTitleMatchMode, 2</code></li>
  <li>要传递消息到隐藏窗口, 请添加此行到脚本: <br><code>DetectHiddenWindows, On</code></li>
</ul>
<p>注: 此技术对某些应用程序无效. 我对 VB 和 Delphi 编写的应用程序只能侥幸. 此技术最适用于 C, C++ 编写的应用程序. 对于 VB 应用程序相同命令的 'LParam' 参数在传递时总是变化的. 对于 Delphi 应用程序... 一些程序的 GUI 甚至不使用 wm_command. 它或许使用鼠标位置和点击吧.</p>
<p>去探索吧.... 并且记得在 AutoHotkey 论坛分享您的经验. 欢迎反馈!</p>
<p>这个指南并不是为新手们准备的(没有冒犯的意思), 因为这些命令被认为是高级功能. 所以如果读完本文, 您还是摸不着头脑, 请忘了它吧.</p>
<p>Rajat</p>
</body>
</html>